home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1998 March / Macworld (1998-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / latex Mode / latexMacros.tcl < prev    next >
Encoding:
Text File  |  1997-12-08  |  50.0 KB  |  1,889 lines  |  [TEXT/ALFA]

  1. #############################################################################
  2. #############################################################################
  3. #
  4. # latexMacros.tcl (called from latex.tcl)
  5. #
  6. #############################################################################
  7. #
  8. # Author:  Tom Scavo <trscavo@syr.edu>
  9. #
  10. #############################################################################
  11. #############################################################################
  12.  
  13. proc latexMacros.tcl {} {}
  14.  
  15. #############################################################################
  16. # Basic Commands
  17. #
  18. #############################################################################
  19.  
  20. #--------------------------------------------------------------------------
  21. # Utilities:
  22. #--------------------------------------------------------------------------
  23.  
  24. # A keyboard-bound method of accessing menu commands.  Takes a list of 
  25. # menu items (i.e., the tail of a 'menu' command), the menu name (the 
  26. # argument of the '-n' switch) , and the name of a menu filter (the 
  27. # argument of the '-p' switch) as input, and displays these items in a 
  28. # list box.  If the chosen item is a menu command (as opposed to a 
  29. # submenu), it is passed to the menu filter; otherwise, 'chooseCommand' 
  30. # recursively calls itself until a menu command is chosen or the cancel 
  31. # button is pressed.
  32. #
  33. proc chooseCommand {menuItems {menuName ""} {menuFilterProc ""} {level 1}} {
  34.     watchCursor
  35.     if { $menuItems == "" } { return }
  36.     # Preprocess the list of menu items:
  37.     foreach    item $menuItems {
  38.         regsub -all {[<!/].} $item {} item
  39.         regsub -all {…}    $item {} item
  40.         lappend    menOut $item
  41.         if { [string match "menu*" $item] } {
  42.             if { [set ind [lsearch $item {-n}]] >= 0 } {
  43.                 lappend    top "[lindex $item [incr    ind]]:"
  44.             }
  45.         } elseif { ![string match "(*" $item] } {
  46.             lappend    top $item
  47.         }
  48.     }
  49.     # Present the menu items to the user:
  50.     set res    [listpick -p "Choose menu command (level $level):" $top]
  51.     # Either execute a command or recurse on a submenu:
  52.     if { [lsearch $menOut $res] >= 0 } {
  53.         # Execute the command via the menu filter, if necessary:
  54.         if { $menuFilterProc == "" } {
  55.             $res
  56.         } else {
  57.             $menuFilterProc $menuName $res
  58.         }
  59.     } else {
  60.         set res [string trimright $res {:}]
  61.         foreach    item $menOut {
  62.             if { [lsearch $item $res] >= 0 } {
  63.                 set menuItems [lindex $item end]
  64.                 # Determine the name of this submenu:
  65.                 if { [set ind [lsearch $item {-n}]] >= 0 } {
  66.                     set menuName [lindex $item [incr ind]]
  67.                 } else {
  68.                     set menuName ""
  69.                 }
  70.                 # Determine the name of the menu filter for this submenu:
  71.                 if { [set ind [lsearch $item {-p}]] >= 0 } {
  72.                     set menuFilterProc [lindex $item [incr ind]]
  73.                 } else {
  74.                     set menuFilterProc ""
  75.                 }
  76.                 return [chooseCommand $menuItems $menuName $menuFilterProc [incr level]]
  77.             }
  78.         }
  79.     }
  80. }
  81.  
  82. # A keyboard-bound method of inserting environments.
  83. #
  84. proc chooseEnvironment {} {
  85.     set env [getEnvironment]
  86.     if { [catch {eval $env}] } {
  87.         doWrapEnvironment $env
  88.     }
  89. }
  90. proc getEnvironment {} {
  91.     if { [catch {prompt "Choose environment:" "abstract" "" \
  92.                  "abstract" \
  93.                  "array" \
  94.                  "center" \
  95.                  "description" \
  96.                  "displaymath" \
  97.                  "document" \
  98.                  "enumerate" \
  99.                  "eqnarray" \
  100.                  "eqnarray*" \
  101.                  "equation" \
  102.                  "figure" \
  103.                  "filecontents" \
  104.                  "flushleft" \
  105.                  "flushright" \
  106.                  "itemize" \
  107.                  "math" \
  108.                  "minipage" \
  109.                  "note" \
  110.                  "overlay" \
  111.                  "quotation" \
  112.                  "quote" \
  113.                  "slide" \
  114.                  "table" \
  115.                  "tabular" \
  116.                  "thebibliography" \
  117.                  "titlepage" \
  118.                  "verbatim" \
  119.                  "verse" \
  120.                 } env] } {
  121.         set env ""
  122.     }
  123.     return $env
  124. }
  125.  
  126.  
  127. #############################################################################
  128. # Paragraph Mode Macros
  129. #
  130. #############################################################################
  131.  
  132. #--------------------------------------------------------------------------
  133. # Documents:
  134. #--------------------------------------------------------------------------
  135.  
  136. proc newLaTeXDocument {} {
  137.     catch {prompt "Choose a documentclass:" "article" "" "article" \
  138.             "report" "book" "letter" "slides"} documentType
  139.     if {$documentType != "cancel"} {
  140.         new -m TeX
  141.         if { [catch {${documentType}Documentclass}] } {
  142.             wrapDocument "$documentType" 
  143.         }
  144.         while { [options] } {}
  145.         message "enter option (or leave blank)"
  146.     }
  147. }
  148.  
  149. proc letterDocumentclass {} {
  150.     set    preamble "\r\\address\{%\r"
  151.     append preamble "    ••    \\\\    % insert your name here\r"
  152.     append preamble "    ••    \\\\    % insert your address here\r"
  153.     append preamble "    ••    \\\\    % insert more address here\r"
  154.     append preamble "    ••          % insert city-state-zip here\r"
  155.     append preamble "\}\r\r"
  156.     append preamble "\\date\{••\}  % optional\r"
  157.     append preamble "\\signature\{••\}\r\r"
  158.     set    body     "\r\\begin\{letter\}\{%\r"
  159.     append body     "    ••    \\\\    % insert addressee's name here\r"
  160.     append body     "    ••    \\\\    % insert addressee's address here\r"
  161.     append body     "    ••    \\\\    % insert more address here\r"
  162.     append body     "    ••          % insert addressee's city-state-zip here\r"
  163.     append body     "\}\r\r"
  164.     append body     "\\opening\{Dear ••,\}\r\r"
  165.     if {[isEmptyFile]} {
  166.         append body "% BODY OF LETTER\r"
  167.         append body "••\r\r"
  168.     } else {
  169.         if {[isSelectionAll]} {
  170.             set text [getSelect]
  171. #             deleteText 0 [maxPos]
  172.             append body "$text\r"
  173.         } else {
  174.             alertnote "nonempty file:  delete text or \'Select All\'\
  175.                 from the Edit menu"
  176.             return
  177.         }
  178.     }
  179.     append body "\\closing\{Sincerely,\}\r\r"
  180.     append body "\\encl\{••\}\r"
  181.     append body "\\cc\{••\}\r\r"
  182.     append body "\\end\{letter\}\r\r"
  183.     insertDocument "letter" $preamble $body
  184.     message "enter option (or leave blank)"
  185. }
  186. proc articleDocumentclass {} {
  187.     if { [wrapDocument "article"] } {
  188.         message "enter option (or leave blank)"
  189.     }
  190. }
  191. proc reportDocumentclass {} {
  192.     if { [wrapDocument "report"] } {
  193.         message "enter option (or leave blank)"
  194.     }
  195. }
  196. proc bookDocumentclass {} {
  197.     if { [wrapDocument "book"] } {
  198.         message "enter option (or leave blank)"
  199.     }
  200. }
  201. proc slidesDocumentclass {} {
  202.     if { [wrapDocument "slides"] } {
  203.         message "enter option (or leave blank)"
  204.     }
  205. }
  206. proc otherDocumentclass {} {
  207.     catch {prompt "What documentclass?" "article"} documentType
  208.     if {$documentType != "cancel"} {
  209.         if { [wrapDocument "$documentType"] } {
  210.             message "enter option (or leave blank)"
  211.         }
  212.     }
  213. }
  214.  
  215. # If an option is inserted, return true; otherwise, return false.
  216. proc options {} {
  217.     set option [getOption]
  218.     if {$option != ""} {
  219.         insertOption $option
  220.         return 1
  221.     }
  222.     return 0
  223. }
  224. proc getOption {} {
  225.     catch {prompt "Choose an option:" "11pt" "" "10pt" "11pt" "12pt" "(-" \
  226.             "letterpaper" "legalpaper" "executivepaper" "a4paper" "a5paper" \
  227.             "b5paper" "(-" "landscape" "(-" "final" "draft" "(-" \
  228.             "oneside" "twoside" "(-" "openright" "openany" "(-" \
  229.             "onecolumn" "twocolumn" "(-" "notitlepage" "titlepage" \
  230.                   "openbib" "(-" "leqno" "(-" "fleqn"} optionName
  231.     if {$optionName != "cancel"} {
  232.         return $optionName
  233.     } else {
  234.         return ""
  235.     }
  236. }
  237. proc insertOption {option} {
  238.     global TeXmodeVars
  239.     set searchString {\\documentclass}
  240.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  241.     if {[llength $searchResult] == 0} {
  242.         if { $TeXmodeVars(searchNoisily) } {beep}
  243.         message "can\'t find \\documentclass"
  244.     } else {
  245.         set nextCharPos [lindex $searchResult 1]
  246.         goto $nextCharPos
  247.         set nextChar [lookAt $nextCharPos]
  248.         if {$nextChar == "\["} {
  249.             forwardChar
  250.             insertText $option
  251.             if {[lookAt [getPos]] != "\]"} {
  252.                 insertText ","
  253.             }
  254.         } elseif {$nextChar == "\{"} {
  255.             insertText "\[$option\]"
  256.         } else {
  257.             alertnote "unrecognizable \\documentclass statement"
  258.         }
  259.     }
  260. }
  261.  
  262. proc insertPackage {package} {
  263.     global TeXmodeVars
  264.     # Check to see if $package is already loaded:
  265.     if { $package != "" } {
  266.         append searchString {^[^%]*\\usepackage\{.*} $package {.*\}}
  267.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  268.         if {[llength $searchResult] != 0} {
  269.             if { $TeXmodeVars(searchNoisily) } {beep}
  270.             message "$package package already loaded"
  271.             return
  272.         }
  273.     }
  274.     # Newlines are allowed in the arguments of \documentclass:
  275.     set searchString {\\documentclass(\[[^][]*\])?{[^{}]*}}
  276.     # Search for \documentclass command:
  277.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  278.     if {[llength $searchResult] == 0} {
  279.         if { $TeXmodeVars(searchNoisily) } {beep}
  280.         message "can't find \\documentclass"
  281.     } else {
  282.         pushPosition
  283.         goto [lindex $searchResult 1]
  284.         set txt "\r\\usepackage\{$package\}"
  285.         insertText $txt
  286.         backwardChar
  287.         message "Press <Ctl .> to return to previous position"
  288.     }
  289. }
  290.  
  291. proc filecontents {} {
  292.     global searchNoisily
  293.     set searchString {\\documentclass}
  294.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  295.     if {[llength $searchResult] == 0} {
  296.         if {$searchNoisily} {beep}
  297.         message "can\'t find \\documentclass"
  298.         return
  299.     } else {
  300.         set prompt "File to be included:"
  301.         if {[catch {getfile $prompt} path]} {
  302.             return
  303.         } else {
  304.             replaceText 0 0 [buildFilecontents $path]
  305.             goto 0
  306.             message "file included"
  307.         }
  308.     }
  309. }
  310. proc filecontentsAll {} {
  311.     global searchNoisily
  312.     watchCursor
  313.     message "locating all input files…"
  314.     set currentWin [win::Current]
  315.     # Is the current window part of TeX fileset?
  316.     set fset [isWindowInFileset $currentWin "tex"]
  317.     if { $fset == "" } {
  318.         set searchString {\\documentclass}
  319.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  320.         if {[llength $searchResult] == 0} {
  321.             if {$searchNoisily} {beep}
  322.             message "can\'t find \\documentclass"
  323.             return
  324.         } else {
  325.             set text [getText 0 [maxPos]]
  326.         }
  327.     } else {
  328.         # Will not handle a base file that is open and dirty:
  329.         set text [buildFilecontents [texFilesetBaseName $fset]]
  330.     }
  331.     set currentDir [file dirname $currentWin]
  332.     set newText [texResolveAll $text $currentDir]
  333.     if { [string length $text] == [string length $newText] } {
  334.         beep
  335.         message "no files to include"
  336.     } else {
  337.         replaceText 0 [maxPos] $newText
  338.         goto 0
  339.         message "all files included"
  340.     }
  341. }
  342. # Takes a LaTeX document string and a path as input, and returns
  343. # a modified document string with all filecontents environments
  344. # prepended.
  345. proc texResolveAll {latexDoc currentDir} {
  346.     global TeXmodeVars
  347.     set pairs [list \
  348.                 {{\\documentclass} {.cls}} {{\\LoadClass} {.cls}} \
  349.                 {{\\include} {.tex}} \
  350.                 {{\\usepackage} {.sty}} {{\\RequirePackage} {.sty}} \
  351.                 {{\\input} {}} \
  352.                 {{\\bibliography} {.bib}} {{\\bibliographystyle} {.bst}} \
  353.               ]
  354.     foreach macro $TeXmodeVars(boxMacroNames) {
  355.         regsub {\*} $macro {\\*} macro
  356.         lappend pairs [list \\\\$macro {}]
  357.     }
  358.     foreach pair $pairs {
  359.         set cmd [car $pair]
  360.         set ext [cadr $pair]
  361.         set searchString $cmd
  362.         append searchString {(\[[^][]*\])?{([^{}]*)}}
  363.         set searchText $latexDoc
  364.         while { [regexp -indices $searchString $searchText mtch dummy theArgs] } {
  365.             set begPos [lindex $theArgs 0]
  366.             set endPos [lindex $theArgs 1]
  367.             set args [string range $searchText $begPos $endPos]
  368.             foreach arg [split $args ,] {
  369.                 if { $cmd == {\\input} && ![string length [file extension $arg]] } {
  370.                     set ext {.tex}
  371.                 }
  372.                 set files [glob -nocomplain $currentDir:$arg*]
  373.                 set filename "$currentDir:$arg$ext"
  374.                 if { [lsearch -exact $files $filename] > -1 } {
  375.                     set tempDoc $latexDoc
  376.                     set latexDoc [buildFilecontents $filename]
  377.                     append latexDoc $tempDoc
  378.                 }
  379.             }
  380.             set searchText [string range $searchText [expr $endPos + 2] end]
  381.         }
  382.     }
  383.     return $latexDoc
  384. }
  385. # Takes a filename as input and returns a filecontents environment 
  386. # based on the contents of that file.  If a second argument is given,
  387. # use that as the argument of the filecontents environment instead
  388. # of the original filename.
  389. proc buildFilecontents {filename {newFilename {}}} {
  390.     set text [readFile $filename]
  391.     # Fix end-of-line characters:
  392.     regsub -all "\xa" $text "\xd" text
  393.     set envName "filecontents"
  394.     if { $newFilename == {} } {
  395.         set envArg "{[file tail $filename]}"
  396.     } else {
  397.         set envArg "{$newFilename}"
  398.     }
  399.     return [buildEnvironment $envName $envArg "$text\r" "\r\r"]
  400. }
  401.  
  402. #--------------------------------------------------------------------------
  403. # Page Layout:
  404. #--------------------------------------------------------------------------
  405.  
  406. proc maketitle {} {
  407.     global searchNoisily
  408.     set searchString {\\document(class|style)(\[.*\])?\{.*\}}
  409.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  410.     if {[llength $searchResult] == 0} {
  411.         if {$searchNoisily} {beep}
  412.         message "can\'t find \\documentclass or \\documentstyle"
  413.     } else {
  414.         set searchPos [lindex $searchResult 1]
  415.         set searchString {\\begin\{document\}}
  416.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString $searchPos]
  417.         if {[llength $searchResult] == 0} {
  418.             if {$searchNoisily} {beep}
  419.             message "can\'t find \\begin\{document\}"
  420.         } else {
  421.             goto [lindex $searchResult 1]
  422.             set currentPos [getPos]
  423.             set txt "\r\r% Definition of title page:"
  424.             append txt "\r\\title\{"
  425.             append txt "\r\t••\r\}"
  426.             append txt "\r\\author\{"
  427.             append txt "\r\t••\t% insert author(s) here"
  428.             append txt "\r\}"
  429.             append txt "\r\\date\{••\}\t% optional"
  430.             append txt "\r\r\\maketitle"
  431.             elec::Insertion $txt
  432.             message "insert title"
  433.         }
  434.     }
  435. }
  436. proc abstract {} { doWrapEnvironment "abstract" }
  437. proc titlepage {} { doWrapEnvironment "titlepage" }
  438. proc getPagestyle {} {
  439.     catch {prompt "Choose a pagestyle:" "plain" "" "plain" "empty" \
  440.                   "headings" "myheadings"} pagestyleName
  441.     if {$pagestyleName != "cancel"} {
  442.         return $pagestyleName
  443.     } else {
  444.         return ""
  445.     }
  446. }
  447. proc pagestyle {} {
  448.     set pagestyleName [getPagestyle]
  449.     if {$pagestyleName != ""} {
  450.         openingCarriageReturn
  451.         insertObject "\\pagestyle\{$pagestyleName\}"
  452.         closingCarriageReturn
  453.     }
  454. }
  455. proc thispagestyle {} {
  456.     set pagestyleName [getPagestyle]
  457.     if {$pagestyleName != ""} {
  458.         openingCarriageReturn
  459.         insertObject "\\thispagestyle\{$pagestyleName\}"
  460.         closingCarriageReturn
  461.     }
  462. }
  463. proc getPagenumberingStyle {} {
  464.     catch {prompt "Choose a pagenumbering style:" "arabic" "" "arabic" \
  465.                   "roman" "Roman" "alph" "Alph"} pagenumberingStyle
  466.     if {$pagenumberingStyle != "cancel"} {
  467.         return $pagenumberingStyle
  468.     } else {
  469.         return ""
  470.     }
  471. }
  472. proc pagenumbering {} {
  473.     set pagenumberingStyle [getPagenumberingStyle]
  474.     if {$pagenumberingStyle != ""} {
  475.         openingCarriageReturn
  476.         insertObject "\\pagenumbering\{$pagenumberingStyle\}"
  477.         closingCarriageReturn
  478.     }
  479. }
  480. proc twocolumn {} {
  481.     openingCarriageReturn
  482.     insertObject "\\twocolumn"
  483.     closingCarriageReturn
  484. }
  485. proc onecolumn {} {
  486.     openingCarriageReturn
  487.     insertObject "\\onecolumn"
  488.     closingCarriageReturn
  489. }
  490.  
  491. #--------------------------------------------------------------------------
  492. # Sectioning:
  493. #--------------------------------------------------------------------------
  494.  
  495. proc sectioning {item} {
  496.     append left [openingCarriageReturn] "\\${item}\{"
  497.     append right "\}"
  498.     if [regexp (part|chapter) $item] {
  499.         append right "\\thispagestyle{empty}"
  500.     }
  501.     append right "••" [closingCarriageReturn]
  502.     if [wrapObject $left $right] {
  503.         message "don't forget the label <Ctl Opt L>"
  504.     } else {
  505.         message "type the [string trim $item {*}] name and don't forget the label <Ctl Opt L>"
  506.     }
  507. }
  508.  
  509. proc appendix {} {
  510.     insertObject "[openingCarriageReturn]\\appendix[closingCarriageReturn]"
  511. }
  512.  
  513. #--------------------------------------------------------------------------
  514. # Text Style:
  515. #--------------------------------------------------------------------------
  516.  
  517. proc emph {} {
  518.     if {[wrapObject "\\emph{" "}••"]} {
  519.         message "selected text has been emphasized"
  520.     } else {
  521.         message "enter text to be emphasized"
  522.     }
  523. }
  524. proc textup {} {
  525.     if {[wrapObject "\\textup{" "}••"]} {
  526.         message "selected text has upright shape"
  527.     } else {
  528.         message "enter text to have upright shape"
  529.     }
  530. }
  531. proc textit {} {
  532.     if {[wrapObject "\\textit{" "}••"]} {
  533.         message "selected text has italic shape"
  534.     } else {
  535.         message "enter text to have italic shape"
  536.     }
  537. }
  538. proc textsl {} {
  539.     if {[wrapObject "\\textsl{" "}••"]} {
  540.         message "selected text has slanted shape"
  541.     } else {
  542.         message "enter text to have slanted shape"
  543.     }
  544. }
  545. proc textsc {} {
  546.     if {[wrapObject "\\textsc{" "}••"]} {
  547.         message "selected text has small caps shape"
  548.     } else {
  549.         message "enter text to have small caps shape"
  550.     }
  551. }
  552. proc textmd {} {
  553.     if {[wrapObject "\\textmd{" "}••"]} {
  554.         message "selected text has been set in medium series"
  555.     } else {
  556.         message "enter text to be set in medium series"
  557.     }
  558. }
  559. proc textbf {} {
  560.     if {[wrapObject "\\textbf{" "}••"]} {
  561.         message "selected text has been set in bold series"
  562.     } else {
  563.         message "enter text to be set in bold series"
  564.     }
  565. }
  566. proc textrm {} {
  567.     if {[wrapObject "\\textrm{" "}••"]} {
  568.         message "selected text has been set with roman family"
  569.     } else {
  570.         message "enter text to be set using roman family"
  571.     }
  572. }
  573. proc textsf {} {
  574.     if {[wrapObject "\\textsf{" "}••"]} {
  575.         message "selected text has been set with sans serif family"
  576.     } else {
  577.         message "enter text to be set using sans serif family"
  578.     }
  579. }
  580. proc texttt {} {
  581.     if {[wrapObject "\\texttt{" "}••"]} {
  582.         message "selected text has been set with typewriter family"
  583.     } else {
  584.         message "enter text to be set using typewriter family"
  585.     }
  586. }
  587. proc textnormal {} {
  588.     if {[wrapObject "\\textnormal{" "}••"]} {
  589.         message "selected text has been set with normal style"
  590.     } else {
  591.         message "enter text to be set using normal style"
  592.     }
  593. }
  594.  
  595. proc em {} {
  596.     if {[wrapObject "{\\em " "}••"]} {
  597.         message "emphasized text set"
  598.     } else {
  599.         message "enter text to be emphasized"
  600.     }
  601. }
  602. proc upshape {} {
  603.     if {[wrapObject "{\\upshape " "}••"]} {
  604.         message "text set in upright shape"
  605.     } else {
  606.         message "enter text to be set in upright shape"
  607.     }
  608. }
  609. proc itshape {} {
  610.     if {[wrapObject "{\\itshape " "}••"]} {
  611.         message "text set in italics shape"
  612.     } else {
  613.         message "enter text to be set in italics shape"
  614.     }
  615. }
  616. proc slshape {} {
  617.     if {[wrapObject "{\\slshape " "}••"]} {
  618.         message "text set in slanted shape"
  619.     } else {
  620.         message "enter text to be set in slanted shape"
  621.     }
  622. }
  623. proc scshape {} {
  624.     if {[wrapObject "{\\scshape " "}••"]} {
  625.         message "text set in small caps shape"
  626.     } else {
  627.         message "enter text to be set in small caps shape"
  628.     }
  629. }
  630. proc mdseries {} {
  631.     if {[wrapObject "{\\mdseries " "}••"]} {
  632.         message "text set in medium series"
  633.     } else {
  634.         message "enter text to be set in medium series"
  635.     }
  636. }
  637. proc bfseries {} {
  638.     if {[wrapObject "{\\bfseries " "}••"]} {
  639.         message "text set in bold series"
  640.     } else {
  641.         message "enter text to be set in bold series"
  642.     }
  643. }
  644. proc rmfamily {} {
  645.     if {[wrapObject "{\\rmfamily " "}••"]} {
  646.         message "text set in roman family"
  647.     } else {
  648.         message "enter text to be set in roman family"
  649.     }
  650. }
  651. proc sffamily {} {
  652.     if {[wrapObject "{\\sffamily " "}••"]} {
  653.         message "text set in sans serif family"
  654.     } else {
  655.         message "enter text to be set in sans serif family"
  656.     }
  657. }
  658. proc ttfamily {} {
  659.     if {[wrapObject "{\\ttfamily " "}••"]} {
  660.         message "text set in typewriter family"
  661.     } else {
  662.         message "enter text to be set in typewriter family"
  663.     }
  664. }
  665. proc normalfont {} {
  666.     if {[wrapObject "{\\normalfont " "}••"]} {
  667.         message "text set in normal style"
  668.     } else {
  669.         message "enter text to be set in normal style"
  670.     }
  671. }
  672.  
  673. #--------------------------------------------------------------------------
  674. # Text Size:
  675. #--------------------------------------------------------------------------
  676.  
  677. proc doTextSize {textSize} {
  678.     if {[wrapObject "{\\$textSize " "}••"]} {
  679.         message "$textSize text set"
  680.     } else {
  681.         message "enter $textSize text"
  682.     }
  683. }
  684.  
  685. #--------------------------------------------------------------------------
  686. # Text Commands:
  687. #--------------------------------------------------------------------------
  688.  
  689. proc textsuperscript {} {
  690.     if {[wrapObject "\\textsuperscript{" "}••"]} {
  691.         message "text superscripted"
  692.     } else {
  693.         message "enter superscripted text"
  694.     }
  695. }
  696. proc textcircled {} {
  697.     if {[wrapObject "\\textcircled{" "}••"]} {
  698.         message "text circled"
  699.     } else {
  700.         message "enter circled text"
  701.     }
  702. }
  703.  
  704. #--------------------------------------------------------------------------
  705. # International:
  706. #--------------------------------------------------------------------------
  707.  
  708. proc {ò} {} {
  709.     if {[wrapObject "\\`{" "}••"]} {
  710.         message "accent set"
  711.     } else {
  712.         message "enter single character"
  713.     }
  714. }
  715. proc {ó} {} {
  716.     if {[wrapObject "\\'{" "}••"]} {
  717.         message "accent set"
  718.     } else {
  719.         message "enter single character"
  720.     }
  721. }
  722. proc {ô} {} {
  723.     if {[wrapObject "\\^{" "}••"]} {
  724.         message "accent set"
  725.     } else {
  726.         message "enter single character"
  727.     }
  728. }
  729. proc {ö} {} {
  730.     if {[wrapObject "\\\"{" "}••"]} {
  731.         message "accent set"
  732.     } else {
  733.         message "enter single character"
  734.     }
  735. }
  736. proc {õ} {} {
  737.     if {[wrapObject "\\~{" "}••"]} {
  738.         message "accent set"
  739.     } else {
  740.         message "enter single character"
  741.     }
  742. }
  743. proc {ç} {} {insertObject "\\c\{c\}"}
  744. proc {Ç} {} {insertObject "\\c\{C\}"}
  745. proc {œ} {} {insertObject "\\oe"}
  746. proc {Œ} {} {insertObject "\\OE"}
  747. proc {æ} {} {insertObject "\\ae"}
  748. proc {Æ} {} {insertObject "\\AE"}
  749. proc {å} {} {insertObject "\\aa"}
  750. proc {Å} {} {insertObject "\\AA"}
  751. proc {ø} {} {insertObject "\\o"}
  752. proc {Ø} {} {insertObject "\\O"}
  753. proc {ss} {} {insertObject "\\ss"}
  754. proc {SS} {} {insertObject "\\SS"}
  755. proc {¿} {} {insertObject "?`"}
  756. proc {¡} {} {insertObject "!`"}
  757.  
  758. #--------------------------------------------------------------------------
  759. # Environments:
  760. #--------------------------------------------------------------------------
  761.  
  762. proc enumerate {} {
  763.     global promptNoisily useStatusBar
  764.     set envName "enumerate"
  765.     if {$promptNoisily && $useStatusBar} {beep}
  766.     catch {sPrompt "$envName:  how many items?" 3} numberItems
  767.     if {$numberItems == "cancel"} {
  768.         return
  769.     } elseif {![isPositiveInteger $numberItems]} {
  770.         beep
  771.         message "invalid input:  unsigned, postive integer required"
  772.         return
  773.     }
  774.     if {$numberItems} {
  775.         set body "\t\\item  ••"
  776.         for {set i 1} {$i < $numberItems} {incr i} {
  777.             append body "\r\r\t\\item  ••"
  778.         }
  779.         append body "\r"
  780.     } else {
  781.         set body "\t••\r"
  782.     }
  783.     if {[insertEnvironment $envName "" $body]} {
  784.         message "type first item"
  785.     }
  786. }
  787. proc itemize {} {
  788.     global promptNoisily useStatusBar
  789.     set envName "itemize"
  790.     if {$promptNoisily && $useStatusBar} {beep}
  791.     catch {sPrompt "$envName:  how many items?" 3} numberItems
  792.     if {$numberItems == "cancel"} {
  793.         return
  794.     } elseif {![isPositiveInteger $numberItems]} {
  795.         beep
  796.         message "invalid input:  unsigned, postive integer required"
  797.         return
  798.     }
  799.     if {$numberItems} {
  800.         set body "\t\\item  ••"
  801.         for {set i 1} {$i < $numberItems} {incr i} {
  802.             append body "\r\r\t\\item  ••"
  803.         }
  804.         append body "\r"
  805.     } else {
  806.         set body "\t••\r"
  807.     }
  808.     if {[insertEnvironment $envName "" $body]} {
  809.         message "type first item"
  810.     }
  811. }
  812. proc description {} {
  813.     global promptNoisily useStatusBar
  814.     set envName "description"
  815.     if {$promptNoisily && $useStatusBar} {beep}
  816.     catch {sPrompt "$envName:  how many items?" 3} numberItems
  817.     if {$numberItems == "cancel"} {
  818.         return
  819.     } elseif {![isPositiveInteger $numberItems]} {
  820.         beep
  821.         message "invalid input:  unsigned, postive integer required"
  822.         return
  823.     }
  824.     if {$numberItems} {
  825.         set body "\t\\item\[••\]  ••"
  826.         for {set i 1} {$i < $numberItems} {incr i} {
  827.             append body "\r\r\t\\item\[••\]  ••"
  828.         }
  829.         append body "\r"
  830.     } else {
  831.         set body "\t••\r"
  832.     }
  833.     if {[insertEnvironment $envName "" $body]} {
  834.         message "type first item label"
  835.     }
  836. }
  837. proc thebibliography {} {
  838.     global promptNoisily useStatusBar
  839.     set envName "thebibliography"
  840.     if {$promptNoisily && $useStatusBar} {beep}
  841.     catch {sPrompt "$envName:  how many items?" 3} numberItems
  842.     if {$numberItems == "cancel"} {
  843.         return
  844.     } elseif {![isPositiveInteger $numberItems]} {
  845.         beep
  846.         message "invalid input:  unsigned, postive integer required"
  847.         return
  848.     }
  849.     set arg "{9}"
  850.     if {$numberItems} {
  851.         if {$numberItems > 9} {set arg "{99}"}
  852.         set body "\t\\bibitem{••}  ••"
  853.         for {set i 1} {$i < $numberItems} {incr i} {
  854.             append body "\r\r\t\\bibitem{••}  ••"
  855.         }
  856.         append body "\r"
  857.     } else {
  858.         set body "\t••\r"
  859.     }
  860.     if {[insertEnvironment $envName $arg $body]} {
  861.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 {(9)+} [getPos]]
  862.         eval select $searchResult
  863.         message "Change the length of the key field?"
  864.     }
  865. }
  866. proc slide {} { doWrapEnvironment "slide" }
  867. proc overlay {} { doWrapEnvironment "overlay" }
  868. proc note {} { doWrapEnvironment "note" }
  869. proc figure {} {
  870.     global TeXmodeVars
  871.     set envName "figure"
  872.     set envArg "tbp"
  873.     set arg "\[$envArg\]"
  874.     set theIndentation [getIndentation [getPos]]
  875.     append arg "\r$theIndentation\t\\centering"
  876.     set body ""
  877.     set macro [car $TeXmodeVars(boxMacroNames)]
  878.     if { $macro != "" } {
  879.         set restOfMacros [cdr $TeXmodeVars(boxMacroNames)]
  880.         if { ![llength $restOfMacros] } {
  881.             append body "\t\\$macro{••}\r"
  882.         } else {
  883.             set cmd [list prompt "Choose a box macro:"]
  884.             lappend cmd $macro "" 
  885.             foreach boxMacroName $TeXmodeVars(boxMacroNames) {
  886.                 lappend cmd $boxMacroName
  887.             }
  888.             catch $cmd macro
  889.             if {$macro == "cancel"} {
  890.                 message "operation canceled"
  891.                 return
  892.             } elseif {$macro == ""} {
  893.                 # do nothing
  894.             } else {
  895.                 append body "\t\\$macro{••}\r"
  896.             }
  897.         }
  898.     }
  899.     append body "\t\\caption{••}\r"
  900.     append body "\t\\label{fig:••}\r"
  901.     if { $macro == "" } {
  902.         wrapEnvironment $envName $arg $body
  903.     } else {
  904.         if {![insertEnvironment $envName $arg $body]} {return}
  905.     }
  906.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 "$envArg" [getPos]]
  907.     eval select $searchResult
  908.     message "Modify this argument?  (t=top; b=bottom; p=page; h=here; !=try harder)"
  909. }
  910. proc table {} {
  911.     set envName "table"
  912.     set envArg "tbp"
  913.     set arg "\[$envArg\]"
  914.     set theIndentation [getIndentation [getPos]]
  915.     append arg "\r$theIndentation\t\\centering"
  916.     # The following statement puts the caption at the top:
  917.     append arg "\r$theIndentation\t\\caption{••}"
  918.     # The following statement puts the caption at the bottom:
  919. #     set body "\t\\caption{••}\r"
  920.     append body "\t\\label{tbl:••}\r"
  921.     wrapEnvironment $envName $arg $body
  922.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 "$envArg" [getPos]]
  923.     eval select $searchResult
  924.     message "Modify this argument?  (t=top; b=bottom; p=page; h=here; !=try harder)"
  925. }
  926. proc buildRow {jmax} {
  927.     set txt "••"
  928.     for {set j 1} {$j < $jmax} {incr j} {
  929.         append txt " & ••"
  930.     }
  931.     return $txt
  932. }
  933. proc tabular {} {
  934.     global promptNoisily useStatusBar
  935.     set envName "tabular"
  936.     if {$promptNoisily && $useStatusBar} {beep}
  937.     catch {sPrompt "$envName:  how many rows?" 3} numberRows
  938.     if {$numberRows == "cancel"} {
  939.         return
  940.     } elseif {![isPositiveInteger $numberRows]} {
  941.         beep
  942.         message "invalid input:  unsigned, postive integer required"
  943.         return
  944.     }
  945.     if {$promptNoisily && $useStatusBar} {beep}
  946.     catch {sPrompt "$envName:  how many columns?" 3} numberCols
  947.     if {$numberCols == "cancel"} {
  948.         return
  949.     } elseif {![isPositiveInteger $numberCols]} {
  950.         beep
  951.         message "invalid input:  unsigned, postive integer required"
  952.         return
  953.     }
  954.     set arg "{|"
  955.     for {set j 1} {$j <= $numberCols} {incr j} {
  956.         append arg "c|"
  957.     }
  958.     append arg "}"
  959.     set body "\t\\hline\r"
  960.     for {set i 1} {$i <= $numberRows} {incr i} {
  961.         append body "\t[buildRow $numberCols]"
  962.         append body "  \\\\\r\t\\hline\r"
  963.     }
  964.     if {[insertEnvironment $envName $arg $body]} {
  965.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 {(c|\|)+} [getPos]]
  966.         select [lindex $searchResult 0] [lindex $searchResult 1]
  967.         message "Modify this argument?"
  968.     }
  969. }
  970. proc verbatim {} { doWrapEnvironment "verbatim" }
  971. proc quote {} { doWrapEnvironment "quote" }
  972. proc quotation {} { doWrapEnvironment "quotation" }
  973. proc verse {} { doWrapEnvironment "verse" }
  974. proc flushleft {} { doWrapEnvironment "flushleft" }
  975. proc center {} { doWrapEnvironment "center" }
  976. proc flushright {} { doWrapEnvironment "flushright" }
  977.  
  978. #--------------------------------------------------------------------------
  979. # Boxes:
  980. #--------------------------------------------------------------------------
  981.  
  982. proc mbox {} {
  983.     if {[wrapObject "\\mbox{" "}••"]} {
  984.         message "mbox set"
  985.     } else {
  986.         message "enter text"
  987.     }
  988. }
  989. proc makebox {} {
  990.     if {[wrapObject "\\makebox\[••\]\[••\]{" "}••"]} {
  991.         message "makebox set; enter the width and position"
  992.     } else {
  993.         message "enter the width and position of the makebox, then the text"
  994.     }
  995. }
  996. proc fbox {} {
  997.     if {[wrapObject "\\fbox{" "}••"]} {
  998.         message "fbox set"
  999.     } else {
  1000.         message "enter text"
  1001.     }
  1002. }
  1003. proc framebox {} {
  1004.     if {[wrapObject "\\framebox\[••\]\[••\]{" "}••"]} {
  1005.         message "framebox set; enter the width and position"
  1006.     } else {
  1007.         message "enter the width and position of the framebox, then the text"
  1008.     }
  1009. }
  1010. proc newsavebox {} {
  1011.     if {[wrapObject "\\newsavebox{" "}••"]} {
  1012.         message "newsavebox defined"
  1013.     } else {
  1014.         message "enter the command name of the sbox or savebox"
  1015.     }
  1016. }
  1017. proc sbox {} {
  1018.     if {[wrapObject "\\sbox{••}{" "}••"]} {
  1019.         message "sbox set; enter the command name"
  1020.     } else {
  1021.         message "enter the command name of the sbox, then the text"
  1022.     }
  1023. }
  1024. proc savebox {} {
  1025.     if {[wrapObject "\\savebox{••}\[••\]\[••\]{" "}••"]} {
  1026.         message "savebox set; enter the command name"
  1027.     } else {
  1028.         message "enter the command name of the savebox"
  1029.     }
  1030. }
  1031. proc usebox {} {
  1032.     if {[wrapObject "\\usebox{" "}••"]} {
  1033.         message "usebox declared"
  1034.     } else {
  1035.         message "enter the command name of the sbox or savebox"
  1036.     }
  1037. }
  1038. proc raisebox {} {
  1039.     if {[wrapObject "\\raisebox{••}\[••\]\[••\]{" "}••"]} {
  1040.         message "raisebox set; enter the displacement"
  1041.     } else {
  1042.         message "enter the displacement of the raisebox"
  1043.     }
  1044. }
  1045. proc parbox {} {
  1046.     if {[wrapObject "\\parbox\[••\]\{••\}{" "}••"]} {
  1047.         message "parbox set; enter the position and width"
  1048.     } else {
  1049.         message "enter the position \[b|c|t\] and width of the parbox, then the text"
  1050.     }
  1051. }
  1052. proc minipage {} {
  1053.     set arg "\[••\]{••}"
  1054.     wrapEnvironment "minipage" $arg ""
  1055.     message "enter the position \[b|c|t\] of the minipage, then the width"
  1056. }
  1057. proc rule {} {
  1058.     insertObject "\\rule\[••\]\{••\}{••}••"
  1059.     message "enter the displacement of the rule, then width and height"
  1060. }
  1061.  
  1062. #--------------------------------------------------------------------------
  1063. # Misc:
  1064. #--------------------------------------------------------------------------
  1065.  
  1066. proc verb {} {
  1067.     if {[wrapObject "\\verb|" "|••"]} {
  1068.         message "verbatim text set"
  1069.     } else {
  1070.         message "enter verbatim text"
  1071.     }
  1072. }
  1073. proc footnote {} {
  1074.     if {[wrapObject "\\footnote{" "}••"]} {
  1075.         message "footnote set"
  1076.     } else {
  1077.         message "enter footnote"
  1078.     }
  1079. }
  1080. proc marginalNote {} {
  1081.     if {[wrapObject "\\marginpar{" "}••"]} {
  1082.         message "marginal note set"
  1083.     } else {
  1084.         message "enter marginal note"
  1085.     }
  1086. }
  1087.  
  1088. proc insertLabel {} {
  1089.     if {[wrapObject "\\label{" "}••"]} {
  1090.         message "label defined"
  1091.     } else {
  1092.         message "enter label"
  1093.     }
  1094. }
  1095. proc ref {} { 
  1096.     if {[wrapObject "\\ref{" "}••" 1]} {
  1097.         message "reference made"
  1098.     } else {
  1099.         message "enter reference label"
  1100.     }
  1101. }
  1102. proc pageref {} { 
  1103.     if {[wrapObject "\\pageref{" "}••" 1]} {
  1104.         message "page reference made"
  1105.     } else {
  1106.         message "enter page reference label"
  1107.     }
  1108. }
  1109. proc cite {} {
  1110.     if {[wrapObject "\\cite{" "}••" 1]} {
  1111.         message "citation made"
  1112.     } else {
  1113.         message "enter citation key"
  1114.     }
  1115. }
  1116. proc nocite {} {
  1117.     if {[wrapObject "\\nocite{" "}••"]} {
  1118.         message "citation added to the list"
  1119.     } else {
  1120.         message "enter citation key"
  1121.     }
  1122. }
  1123.  
  1124. # Insert an \item or a \bibitem, depending on the context.
  1125. proc insertItem {} {
  1126.     set command [eval getText [searchEnvironment]]
  1127.     set environment [extractCommandArg $command]
  1128.     switch $environment {
  1129.         "itemize" {
  1130.             set text "\\item  ••"
  1131.         }
  1132.         "enumerate" {
  1133.             set text "\\item  ••"
  1134.         }
  1135.         "description" {
  1136.             set text "\\item\[••\]  ••"
  1137.         }
  1138.         "thebibliography" {
  1139.             set text "\\bibitem{••}  ••"
  1140.         }
  1141.         default {
  1142.             beep
  1143.             message "insertItem: cursor in $environment environment"
  1144.             return
  1145.         }
  1146.     }
  1147.     set pos [getPos]
  1148.     # Indentation should mirror that of an existing \item
  1149.     # (if it exists)
  1150.     elec::Insertion [openingCarriageReturn]$text
  1151. }
  1152.  
  1153. proc quotes {} {
  1154.     if {[wrapObject "`" "'••"]} {
  1155.         message "text quoted"
  1156.     } else {
  1157.         message "enter text"
  1158.     }
  1159. }
  1160. proc dblQuotes {} {
  1161.     if {[wrapObject "``" "''••"]} {
  1162.         message "text double quoted"
  1163.     } else {
  1164.         message "enter text"
  1165.     }
  1166. }
  1167.  
  1168. proc texLogo {} {insertObject "\\TeX"}
  1169. proc latexLogo {} {insertObject "\\LaTeX"}
  1170. proc latex2eLogo {} {insertObject "\\LaTeXe"}
  1171. proc today {} {insertObject "\\today"}
  1172.  
  1173. proc dag {} {insertObject "\\dag"}
  1174. proc ddag {} {insertObject "\\ddag"}
  1175. proc sectionMark {} {insertObject "\\S"}
  1176. proc paragraphMark {} {insertObject "\\P"}
  1177. proc copyright {} {insertObject "\\copyright"}
  1178. proc pounds {} {insertObject "\\pounds"}
  1179.  
  1180.  
  1181. #############################################################################
  1182. # Math Mode Macros
  1183. #
  1184. #############################################################################
  1185.  
  1186. #--------------------------------------------------------------------------
  1187. # Math Modes:
  1188. #--------------------------------------------------------------------------
  1189.  
  1190. proc texMath {} {
  1191.     checkMathMode "texMath" 0
  1192.     if {[wrapObject "$" "$••"]} {
  1193.         message "formula set"
  1194.     } else {
  1195.         message "enter formula"
  1196.     }
  1197. }
  1198. proc texDisplaymath {} {
  1199.     checkMathMode "texDisplaymath" 0
  1200.     if {[wrapObject "$$" "$$••"]} {
  1201.         message "displayed formula set"
  1202.     } else {
  1203.         message "enter displayed formula"
  1204.     }
  1205. }
  1206. proc latexMath {} {
  1207.     checkMathMode "latexMath" 0
  1208.     if {[wrapObject "\\( " " \\)••"]} {
  1209.         message "formula set"
  1210.     } else {
  1211.         message "enter formula"
  1212.     }
  1213. }
  1214. proc latexDisplaymath {} {
  1215.     checkMathMode "latexDisplaymath" 0
  1216.     if {[wrapObject "\\\[ " " \\\]••"]} {
  1217.         message "displayed formula set"
  1218.     } else {
  1219.         message "enter displayed formula"
  1220.     }
  1221. }
  1222.  
  1223. #--------------------------------------------------------------------------
  1224. # Math Style:
  1225. #--------------------------------------------------------------------------
  1226.  
  1227. proc doMathStyle {mathStyle description} {
  1228.     checkMathMode "$mathStyle" 1
  1229.     if {[wrapObject "\\$mathStyle{" "}••"]} {
  1230.         message "selected text is $description"
  1231.     } else {
  1232.         message "enter text to be $description"
  1233.     }
  1234. }
  1235. proc doUppercaseMathStyle {mathStyle description} {
  1236.     checkMathMode "$mathStyle" 1
  1237.     # Allow upper-case alphabetic arguments only:
  1238.     if {[isSelection] && (![isUppercase] || ![isAlphabetic])} {
  1239.         beep
  1240.         alertnote "argument to \\$mathStyle must be UPPERCASE alphabetic"
  1241.         return
  1242.     }
  1243.     if {[wrapObject "\\$mathStyle{" "}••"]} {
  1244.         message "selected text is $description"
  1245.     } else {
  1246.         message "enter text to be $description (UPPERCASE letters only)"
  1247.     }
  1248. }
  1249.  
  1250. #--------------------------------------------------------------------------
  1251. # Math Environments:
  1252. #--------------------------------------------------------------------------
  1253.  
  1254. proc math {} { checkMathMode "math" 0; doWrapEnvironment "math" }
  1255. proc displaymath {} {
  1256.     checkMathMode "displaymath" 0
  1257.     global TeXmodeVars
  1258.     if { $TeXmodeVars(useBrackets) } {
  1259.         doWrapStructure {\[} {} {\]}
  1260.     } else {
  1261.         doWrapEnvironment "displaymath"
  1262.     }
  1263. }
  1264. proc equation {} {
  1265.     checkMathMode "equation" 0
  1266.     set envName "equation"
  1267.     set body "\t\\label{eq:••}\r"
  1268.     if {[wrapEnvironment $envName "" $body]} {
  1269.         set msgText "equation wrapped"
  1270.     } else {
  1271.         set msgText "enter equation"
  1272.     }
  1273.     message $msgText
  1274. }
  1275. proc eqnarrayStar {} {
  1276.     global promptNoisily useStatusBar
  1277.     checkMathMode "eqnarrayStar" 0
  1278.     set envName "eqnarray*"
  1279.     if {$promptNoisily && $useStatusBar} {beep}
  1280.     catch {sPrompt "$envName:  how many rows?" 3} numberRows
  1281.     if {$numberRows == "cancel"} {
  1282.         return
  1283.     } elseif {![isPositiveInteger $numberRows]} {
  1284.         beep
  1285.         message "invalid input:  unsigned, postive integer required"
  1286.         return
  1287.     }
  1288.     set row "\t[buildRow 3]"
  1289.     for {set i 1} {$i < $numberRows} {incr i} {
  1290.         append body $row
  1291.         append body "  \\\\\r"
  1292.     }
  1293.     append body $row
  1294.     append body "\r"
  1295.     if {[insertEnvironment $envName "" $body]} {
  1296.         message "type first item"
  1297.     }
  1298. }
  1299. proc eqnarray {} {
  1300.     global promptNoisily useStatusBar
  1301.     checkMathMode "eqnarray" 0
  1302.     set envName "eqnarray"
  1303.     if {$promptNoisily && $useStatusBar} {beep}
  1304.     catch {sPrompt "$envName:  how many rows?" 3} numberRows
  1305.     if {$numberRows == "cancel"} {
  1306.         return
  1307.     } elseif {![isPositiveInteger $numberRows]} {
  1308.         beep
  1309.         message "invalid input:  unsigned, postive integer required"
  1310.         return
  1311.     }
  1312.     set row "\t[buildRow 3]\r\t\\label{eq:••}"
  1313.     for {set i 1} {$i < $numberRows} {incr i} {
  1314.         append body $row
  1315.         append body "  \\\\\r"
  1316.     }
  1317.     append body $row
  1318.     append body "\r"
  1319.     if {[insertEnvironment $envName "" $body]} {
  1320.         message "type first item"
  1321.     }
  1322. }
  1323. proc myArray {} {
  1324.     global promptNoisily useStatusBar
  1325.     checkMathMode "myArray" 1
  1326.     set envName "array"
  1327.     if {$promptNoisily && $useStatusBar} {beep}
  1328.     catch {sPrompt "$envName:  how many rows?" 3} numberRows
  1329.     if {$numberRows == "cancel"} {
  1330.         return
  1331.     } elseif {![isPositiveInteger $numberRows]} {
  1332.         beep
  1333.         message "invalid input:  unsigned, postive integer required"
  1334.         return
  1335.     }
  1336.     if {$promptNoisily && $useStatusBar} {beep}
  1337.     catch {sPrompt "$envName:  how many columns?" 3} numberCols
  1338.     if {$numberCols == "cancel"} {
  1339.         return
  1340.     } elseif {![isPositiveInteger $numberCols]} {
  1341.         beep
  1342.         message "invalid input:  unsigned, postive integer required"
  1343.         return
  1344.     }
  1345.     set arg "{"
  1346.     for {set j 1} {$j <= $numberCols} {incr j} {
  1347.         append arg "c"
  1348.     }
  1349.     append arg "}"
  1350.     set row "\t[buildRow $numberCols]"
  1351.     for {set i 1} {$i < $numberRows} {incr i} {
  1352.         append body $row
  1353.         append body "  \\\\\r"
  1354.     }
  1355.     append body $row
  1356.     append body "\r"
  1357.     if {[insertEnvironment $envName $arg $body]} {
  1358.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 {c+} [getPos]]
  1359.         select [lindex $searchResult 0] [lindex $searchResult 1]
  1360.         message "Modify this argument?  (c=center; l=left; r=right; p{width}; a{text})"
  1361.     }
  1362. }
  1363.  
  1364. #--------------------------------------------------------------------------
  1365. # Formulas:
  1366. #--------------------------------------------------------------------------
  1367.  
  1368. proc subscript {} {
  1369.     checkMathMode "subscript" 1
  1370.     if {[wrapObject "_{" "}••"]} {
  1371.         message "subscript set"
  1372.     } else {
  1373.         message "enter subscript"
  1374.     }
  1375. }
  1376. proc superscript {} {
  1377.     checkMathMode "superscript" 1
  1378.     if {[wrapObject "^{" "}••"]} {
  1379.         message "superscript set"
  1380.     } else {
  1381.         message "enter superscript"
  1382.     }
  1383. }
  1384. proc fraction {} {
  1385.     checkMathMode "fraction" 1
  1386.     set currentPos [getPos]
  1387.     if {[isSelection]} {
  1388.         set selection [getSelect]
  1389.         set args [split $selection /]
  1390.         set len [llength $args]
  1391.         deleteText $currentPos [selEnd]
  1392.         if {$len == 1} {
  1393.             insertObject "\\frac{$selection}{••}••"
  1394.             message "enter denominator"
  1395.         } else {
  1396.             set firstArg [lindex $args 0]
  1397.             set restArgs [lrange $args 1 [expr $len-1]]
  1398.             insertObject "\\frac{$firstArg}{[join $restArgs /]}"
  1399.             if {$len > 2} {message "beware of multiple /"}
  1400.         }
  1401.     } else {
  1402.         insertObject "\\frac{••}{••}••"
  1403.         message "enter numerator"
  1404.     }
  1405. }
  1406. proc squareRoot {} {
  1407.     checkMathMode "squareRoot" 1
  1408.     if {[wrapObject "\\sqrt{" "}••"]} {
  1409.         message "square root set"
  1410.     } else {
  1411.         message "enter formula"
  1412.     }
  1413. }
  1414. proc nthRoot {} {
  1415.     checkMathMode "nthRoot" 1
  1416.     if {[wrapObject "\\sqrt\[••\]{" "}••"]} {
  1417.         message "enter root"
  1418.     } else {
  1419.         message "enter root, then formula"
  1420.     }
  1421. }
  1422. proc oneParameter {} {
  1423.     checkMathMode "oneParameter" 1
  1424.     if {[wrapObject "\\••{" "}••"]} {
  1425.         message "enter command name"
  1426.     } else {
  1427.         message "enter command name, press <Tab>, enter argument"
  1428.     }
  1429. }
  1430. proc twoParameters {} {
  1431.     checkMathMode "twoParameters" 1
  1432.     if {[wrapObject "\\••{" "}{••}••"]} {
  1433.         message "enter command name"
  1434.     } else {
  1435.         message "enter command name, press <Tab>, enter argument, etc."
  1436.     }
  1437. }
  1438.  
  1439. #--------------------------------------------------------------------------
  1440. # Large Ops:
  1441. #--------------------------------------------------------------------------
  1442.  
  1443. proc insertLargeOp {commandName} {
  1444.     checkMathMode "$commandName" 1
  1445.     insertObject "\\$commandName\_{••}^{••}••"
  1446. }
  1447. proc sum {} {insertLargeOp "sum"}
  1448. proc prod {} {insertLargeOp "prod"}
  1449. proc coprod {} {insertLargeOp "coprod"}
  1450. proc int {} {insertLargeOp "int"}
  1451. proc oint {} {insertLargeOp "oint"}
  1452. proc bigcap {} {insertLargeOp "bigcap"}
  1453. proc bigcup {} {insertLargeOp "bigcup"}
  1454. proc bigsqcup {} {insertLargeOp "bigsqcup"}
  1455. proc bigvee {} {insertLargeOp "bigvee"}
  1456. proc bigwedge {} {insertLargeOp "bigwedge"}
  1457. proc bigodot {} {insertLargeOp "bigodot"}
  1458. proc bigotimes {} {insertLargeOp "bigotimes"}
  1459. proc bigoplus {} {insertLargeOp "bigoplus"}
  1460. proc biguplus {} {insertLargeOp "biguplus"}
  1461.  
  1462. #--------------------------------------------------------------------------
  1463. # Delimiters:
  1464. #--------------------------------------------------------------------------
  1465.  
  1466. proc delimitObject {leftDelim rightDelim} {
  1467.     if {[wrapObject $leftDelim $rightDelim]} {
  1468.         message "formula delimited"
  1469.     } else {
  1470.         message "enter formula"
  1471.     }
  1472. }
  1473. proc parentheses {} { checkMathMode "parentheses" 1; delimitObject "(" ")••" }
  1474. proc brackets {} { checkMathMode "brackets" 1; delimitObject "\[" "\]••" }
  1475. proc braces {} { checkMathMode "braces" 1; delimitObject "\\\{" "\\\}••" }
  1476. proc absoluteValue {} { checkMathMode "absoluteValue" 1; delimitObject "|" "|••" }
  1477. proc getDelims {} {
  1478.     catch {prompt "Choose delimiters:" "parentheses" "" "parentheses" \
  1479.                   "brackets" "braces" "angle brackets" "vertical bars" \
  1480.                   "double bars" "ceiling" "floor"} delimType
  1481.     if {$delimType != "cancel"} {
  1482.         switch $delimType {
  1483.             "parentheses" {
  1484.                 set leftDelim "("
  1485.                 set rightDelim ")"
  1486.             }
  1487.             "brackets" {
  1488.                 set leftDelim "\["
  1489.                 set rightDelim "\]"
  1490.             }
  1491.             "braces" {
  1492.                 set leftDelim "\\\{"
  1493.                 set rightDelim "\\\}"
  1494.             }
  1495.             "vertical bars" {
  1496.                 set leftDelim "|"
  1497.                 set rightDelim "|"
  1498.             }
  1499.             "double bars" {
  1500.                 set leftDelim "\\|"
  1501.                 set rightDelim "\\|"
  1502.             }
  1503.             "angle brackets" {
  1504.                 set leftDelim "\\langle"
  1505.                 set rightDelim "\\rangle"
  1506.             }
  1507.             "ceiling" {
  1508.                 set leftDelim "\\lceil"
  1509.                 set rightDelim "\\rceil"
  1510.             }
  1511.             "floor" {
  1512.                 set leftDelim "\\lfloor"
  1513.                 set rightDelim "\\rfloor"
  1514.             }
  1515.             default {
  1516.                 alertnote "\"$delimType\" not recognized"
  1517.                 return ""
  1518.             }
  1519.         }
  1520.         return [list $leftDelim $rightDelim]
  1521.     } else {return ""}
  1522. }
  1523. proc otherDelims {} {
  1524.     checkMathMode "otherDelims" 1
  1525.     set delims [getDelims]
  1526.     if {$delims != ""} {
  1527.         set leftDelim [lindex $delims 0]
  1528.         set rightDelim [lindex $delims 1]
  1529.         delimitObject "$leftDelim" "$rightDelim••"
  1530.     }
  1531. }
  1532. proc {half-openInterval} {} {
  1533.     checkMathMode "half-openInterval" 1; delimitObject "(" "\]••"
  1534. }
  1535. proc {half-closedInterval} {} {
  1536.     checkMathMode "half-closedInterval" 1; delimitObject "\[" ")••"
  1537. }
  1538.  
  1539. proc insertBigDelims {leftDelim rightDelim isMultiline} {
  1540.     checkMathMode "insertBigDelims" 1
  1541.     if {$isMultiline} {
  1542.         doWrapStructure $leftDelim "" $rightDelim
  1543.     } else {
  1544.         if { [wrapObject $leftDelim $rightDelim] } {
  1545.             message "formula delimited"
  1546.         } else {
  1547.             message "enter formula"
  1548.         }
  1549.     }
  1550. }
  1551. proc bigParens {} {
  1552.     checkMathMode "bigParens" 1; insertBigDelims "\\left(" "\\right)••" 0
  1553. }
  1554. proc multiBigParens {} {
  1555.     checkMathMode "multiBigParens" 1; insertBigDelims "\\left(" "\\right)••" 1
  1556. }
  1557. proc bigBrackets {} {
  1558.     checkMathMode "bigBrackets" 1; insertBigDelims "\\left\[" "\\right\]••" 0
  1559. }
  1560. proc multiBigBrackets {} {
  1561.     checkMathMode "multiBigBrackets" 1; insertBigDelims "\\left\[" "\\right\]••" 1
  1562. }
  1563. proc bigBraces {} {
  1564.     checkMathMode "bigBraces" 1; insertBigDelims "\\left\\\{" "\\right\\\}••" 0
  1565. }
  1566. proc multiBigBraces {} {
  1567.     checkMathMode "multiBigBraces" 1; insertBigDelims "\\left\\\{" "\\right\\\}••" 1
  1568. }
  1569. proc bigAbsValue {} {
  1570.     checkMathMode "bigAbsValue" 1; insertBigDelims "\\left|" "\\right|••" 0
  1571. }
  1572. proc multiBigAbsValue {} {
  1573.     checkMathMode "multiBigAbsValue" 1; insertBigDelims "\\left|" "\\right|••" 1
  1574. }
  1575. proc doOtherBigDelims {name isMultiline} {
  1576.     checkMathMode $name 1
  1577.     set delims [getDelims]
  1578.     if {$delims != ""} {
  1579.         append leftDelim "\\left" [lindex $delims 0]
  1580.         append rightDelim "\\right" [lindex $delims 1]
  1581.         insertBigDelims "$leftDelim" "$rightDelim••" $isMultiline
  1582.     }
  1583. }
  1584. proc otherBigDelims {} {
  1585.     doOtherBigDelims "otherBigDelims" 0
  1586. }
  1587. proc otherMultiBigDelims {} {
  1588.     doOtherBigDelims "otherMultiBigDelims" 1
  1589. }
  1590. proc bigLeftBrace {} {
  1591.     checkMathMode "bigLeftBrace" 1
  1592.     insertBigDelims "\\left\\\{" "\\right.••" 0
  1593. }
  1594. proc multiBigLeftBrace {} {
  1595.     checkMathMode "multiBigLeftBrace" 1
  1596.     insertBigDelims "\\left\\\{" "\\right.••" 1
  1597. }
  1598. proc doOtherMixedBigDelims {name isMultiline} {
  1599.     checkMathMode $name 1
  1600.     catch {prompt "Choose LEFT delimiter:" "parenthesis" "" "parenthesis" \
  1601.                   "bracket" "brace" "vertical bar" "double bar" \
  1602.                   "angle bracket" "ceiling" "floor" "slash" "backslash" \
  1603.                   "none"} delimType
  1604.     if {$delimType != "cancel"} {
  1605.         switch $delimType {
  1606.             "parenthesis" {set leftDelim "("}
  1607.             "bracket" {set leftDelim "\["}
  1608.             "brace" {set leftDelim "\\\{"}
  1609.             "vertical bar" {set leftDelim "|"}
  1610.             "double bar" {set leftDelim "\\|"}
  1611.             "angle bracket" {set leftDelim "\\langle"}
  1612.             "ceiling" {set leftDelim "\\lceil"}
  1613.             "floor" {set leftDelim "\\lfloor"}
  1614.             "slash" {set leftDelim "/"}
  1615.             "backslash" {set leftDelim "\\backslash"}
  1616.             "none" {set leftDelim "."}
  1617.             default {
  1618.                 alertnote "\"$delimType\" not recognized"
  1619.                 return
  1620.             }
  1621.         }
  1622.         catch {prompt "Choose RIGHT delimiter:" "parenthesis" "" "parenthesis" \
  1623.                       "bracket" "brace" "vertical bar" "double bar" \
  1624.                       "angle bracket" "ceiling" "floor" "slash" "backslash" \
  1625.                       "none"} delimType
  1626.         if {$delimType != "cancel"} {
  1627.             switch $delimType {
  1628.                 "parenthesis" {set rightDelim ")"}
  1629.                 "bracket" {set rightDelim "\]"}
  1630.                 "brace" {set rightDelim "\\\}"}
  1631.                 "vertical bar" {set rightDelim "|"}
  1632.                 "double bar" {set rightDelim "\\|"}
  1633.                 "angle bracket" {set rightDelim "\\rangle"}
  1634.                 "ceiling" {set rightDelim "\\rceil"}
  1635.                 "floor" {set rightDelim "\\rfloor"}
  1636.                 "slash" {set rightDelim "/"}
  1637.                 "backslash" {set rightDelim "\\backslash"}
  1638.                 "none" {set rightDelim "."}
  1639.                 default {
  1640.                     alertnote "\"$delimType\" not recognized"
  1641.                     return
  1642.                 }
  1643.             }
  1644.             insertBigDelims "\\left$leftDelim" "\\right$rightDelim••" $isMultiline
  1645.         }
  1646.     }
  1647. }
  1648. proc otherMixedBigDelims {} {
  1649.     doOtherMixedBigDelims "otherMixedBigDelims" 0
  1650. }
  1651. proc otherMultiMixedBigDelims {} {
  1652.     doOtherMixedBigDelims "otherMultiMixedBigDelims" 1
  1653. }
  1654.  
  1655. #--------------------------------------------------------------------------
  1656. # Accents:
  1657. #--------------------------------------------------------------------------
  1658.  
  1659. proc acute {} {
  1660.     checkMathMode "acute" 1
  1661.     if {[isSelection] > 1} {
  1662.         alertnote "Warning: only a single character may be accented!"
  1663.     }
  1664.     if {[wrapObject "\\acute{" "}••"]} {
  1665.         message "accent set"
  1666.     } else {
  1667.         message "enter one character"
  1668.     }
  1669. }
  1670. proc bar {} {
  1671.     checkMathMode "bar" 1
  1672.     if {[isSelection] > 1} {
  1673.         alertnote "Warning: only a single character may be accented!"
  1674.     }
  1675.     if {[wrapObject "\\bar{" "}••"]} {
  1676.         message "accent set"
  1677.     } else {
  1678.         message "enter one character"
  1679.     }
  1680. }
  1681. proc breve {} {
  1682.     checkMathMode "breve" 1
  1683.     if {[isSelection] > 1} {
  1684.         alertnote "Warning: only a single character may be accented!"
  1685.     }
  1686.     if {[wrapObject "\\breve{" "}••"]} {
  1687.         message "accent set"
  1688.     } else {
  1689.         message "enter one character"
  1690.     }
  1691. }
  1692. proc check {} {
  1693.     checkMathMode "check" 1
  1694.     if {[isSelection] > 1} {
  1695.         alertnote "Warning: only a single character may be accented!"
  1696.     }
  1697.     if {[wrapObject "\\check{" "}••"]} {
  1698.         message "accent set"
  1699.     } else {
  1700.         message "enter one character"
  1701.     }
  1702. }
  1703. proc dot {} {
  1704.     checkMathMode "dot" 1
  1705.     if {[isSelection] > 1} {
  1706.         alertnote "Warning: only a single character may be accented!"
  1707.     }
  1708.     if {[wrapObject "\\dot{" "}••"]} {
  1709.         message "accent set"
  1710.     } else {
  1711.         message "enter one character"
  1712.     }
  1713. }
  1714. proc ddot {} {
  1715.     checkMathMode "ddot" 1
  1716.     if {[isSelection] > 1} {
  1717.         alertnote "Warning: only a single character may be accented!"
  1718.     }
  1719.     if {[wrapObject "\\ddot{" "}••"]} {
  1720.         message "accent set"
  1721.     } else {
  1722.         message "enter one character"
  1723.     }
  1724. }
  1725. proc grave {} {
  1726.     checkMathMode "grave" 1
  1727.     if {[isSelection] > 1} {
  1728.         alertnote "Warning: only a single character may be accented!"
  1729.     }
  1730.     if {[wrapObject "\\grave{" "}••"]} {
  1731.         message "accent set"
  1732.     } else {
  1733.         message "enter one character"
  1734.     }
  1735. }
  1736. proc hat {} {
  1737.     checkMathMode "hat" 1
  1738.     if {[isSelection] > 1} {
  1739.         alertnote "Warning: only a single character may be accented!"
  1740.     }
  1741.     if {[wrapObject "\\hat{" "}••"]} {
  1742.         message "accent set"
  1743.     } else {
  1744.         message "enter one character"
  1745.     }
  1746. }
  1747. proc tilde {} {
  1748.     checkMathMode "tilde" 1
  1749.     if {[isSelection] > 1} {
  1750.         alertnote "Warning: only a single character may be accented!"
  1751.     }
  1752.     if {[wrapObject "\\tilde{" "}••"]} {
  1753.         message "accent set"
  1754.     } else {
  1755.         message "enter one character"
  1756.     }
  1757. }
  1758. proc vec {} {
  1759.     checkMathMode "vec" 1
  1760.     if {[isSelection] > 1} {
  1761.         alertnote "Warning: only a single character may be accented!"
  1762.     }
  1763.     if {[wrapObject "\\vec{" "}••"]} {
  1764.         message "accent set"
  1765.     } else {
  1766.         message "enter one character"
  1767.     }
  1768. }
  1769.  
  1770. proc widehat {} {
  1771.     checkMathMode "widehat" 1
  1772.     if {[isSelection] > 3} {
  1773.         alertnote "Warning: only a few characters may be accented!"
  1774.     }
  1775.     if {[wrapObject "\\widehat{" "}••"]} {
  1776.         message "accent set"
  1777.     } else {
  1778.         message "enter a few characters"
  1779.     }
  1780. }
  1781. proc widetilde {} {
  1782.     checkMathMode "widetilde" 1
  1783.     if {[isSelection] > 3} {
  1784.         alertnote "Warning: only a few characters may be accented!"
  1785.     }
  1786.     if {[wrapObject "\\widetilde{" "}••"]} {
  1787.         message "accent set"
  1788.     } else {
  1789.         message "enter a few characters"
  1790.     }
  1791. }
  1792.  
  1793.  
  1794. #--------------------------------------------------------------------------
  1795. # Grouping:
  1796. #--------------------------------------------------------------------------
  1797.  
  1798. proc underline {} {
  1799.     checkMathMode "underline" 1
  1800.     if {[wrapObject "\\underline{" "}••"]} {
  1801.         message "selection underlined"
  1802.     } else {
  1803.         message "enter text"
  1804.     }
  1805. }
  1806. proc overline {} {
  1807.     checkMathMode "overline" 1
  1808.     if {[wrapObject "\\overline{" "}••"]} {
  1809.         message "selection overlined"
  1810.     } else {
  1811.         message "enter text"
  1812.     }
  1813. }
  1814. proc underbrace {} {
  1815.     checkMathMode "underbrace" 1
  1816.     if {[wrapObject "\\underbrace{" "}••"]} {
  1817.         message "selection underbraced"
  1818.     } else {
  1819.         message "enter text"
  1820.     }
  1821. }
  1822. proc overbrace {} {
  1823.     checkMathMode "overbrace" 1
  1824.     if {[wrapObject "\\overbrace{" "}••"]} {
  1825.         message "selection overbraced"
  1826.     } else {
  1827.         message "enter text"
  1828.     }
  1829. }
  1830. proc overrightarrow {} {
  1831.     checkMathMode "overrightarrow" 1
  1832.     if {[wrapObject "\\overrightarrow{" "}••"]} {
  1833.         message "selection overrightarrowed"
  1834.     } else {
  1835.         message "enter text"
  1836.     }
  1837. }
  1838. proc overleftarrow {} {
  1839.     checkMathMode "overleftarrow" 1
  1840.     if {[wrapObject "\\overleftarrow{" "}••"]} {
  1841.         message "selection overleftarrowed"
  1842.     } else {
  1843.         message "enter text"
  1844.     }
  1845. }
  1846. proc stackrel {} {
  1847.     checkMathMode "stackrel" 1
  1848.     set currentPos [getPos]
  1849.     if {[insertObject "\\stackrel{••}{••}••"]} {
  1850.         message "1st arg scriptstyle"
  1851.     }
  1852. }
  1853.  
  1854. #--------------------------------------------------------------------------
  1855. # Spacing:
  1856. #--------------------------------------------------------------------------
  1857.  
  1858. proc negThin {} {checkMathMode "negThin" 1; insertObject "\\!"}
  1859. proc thin {} {checkMathMode "thin" 1; insertObject "\\,"}
  1860. proc medium {} {checkMathMode "medium" 1; insertObject "\\:"}
  1861. proc thick {} {checkMathMode "thick" 1; insertObject "\\;"}
  1862. proc quad {} {checkMathMode "quad" 1; insertObject "\\quad"}
  1863. proc qquad {} {checkMathMode "qquad" 1; insertObject "\\qquad"}
  1864. proc hspace {} {
  1865.     checkMathMode "hspace" 1
  1866.     if {[wrapObject "\\hspace{" "}••"]} {
  1867.         message "spacing set"
  1868.     } else {
  1869.         message "enter the desired horizontal spacing"
  1870.     }
  1871. }
  1872. proc vspace {} {
  1873.     checkMathMode "vspace" 1
  1874.     if {[wrapObject "\\vspace{" "}••"]} {
  1875.         message "spacing set"
  1876.     } else {
  1877.         message "enter the desired horizontal spacing"
  1878.     }
  1879. }
  1880. proc hfill {} {checkMathMode "hfill" 1; insertObject "\\hfill"}
  1881. proc vfill {} {checkMathMode "vfill" 1; insertObject "\\vfill"}
  1882. proc smallskip {} {checkMathMode "smallskip" 1; insertObject "\\smallskip"}
  1883. proc medskip {} {checkMathMode "medskip" 1; insertObject "\\medskip"}
  1884. proc bigskip {} {checkMathMode "bigskip" 1; insertObject "\\bigskip"}
  1885.  
  1886.